1   /**
2    * Copyright (C) 2006-2019 INRIA and contributors
3    *
4    * Spoon is available either under the terms of the MIT License (see LICENSE-MIT.txt) of the Cecill-C License (see LICENSE-CECILL-C.txt). You as the user are entitled to choose the terms under which to adopt Spoon.
5    */
6   package spoon.reflect.visitor;
7   
8   import spoon.reflect.declaration.CtCompilationUnit;
9   import spoon.reflect.declaration.CtElement;
10  import spoon.reflect.path.CtRole;
11  import spoon.reflect.visitor.chain.CtScannerListener;
12  import spoon.reflect.visitor.chain.ScanningMode;
13  
14  import java.util.ArrayList;
15  import java.util.Collection;
16  import java.util.Map;
17  
18  /**
19   * Extends {@link CtScanner}, to support early termination of scanning process and scan listeners.
20   * It is useful when your algorithm is searching for a specific node only.
21   * In this case, you can call {@link #terminate()}, which ensures that no more AST nodes are visited,
22   *<br>
23   * It is possible to register an implementation of {@link CtScannerListener},
24   * whose {@link CtScannerListener#enter(CtElement)}/{@link CtScannerListener#exit(CtElement)}
25   * methods are called before/after each AST node is visited.<br>
26   *
27   * @param <T> the type of the result produced by this scanner.
28   */
29  public class EarlyTerminatingScanner<T> extends CtScanner {
30  
31  	private boolean terminate = false;
32  	private T result;
33  	private CtScannerListener listener;
34  	protected CtRole scannedRole;
35  	private boolean visitCompilationUnitContent = false;
36  
37  	protected void terminate() {
38  		terminate = true;
39  	}
40  
41  	protected boolean isTerminated() {
42  		return terminate;
43  	}
44  
45  	protected void setResult(T result) {
46  		this.result = result;
47  	}
48  
49  	/**
50  	 * @return the result of scanning - the value, which was stored by a previous call of {@link #setResult(Object)}
51  	 */
52  	public T getResult() {
53  		return result;
54  	}
55  
56  	/**
57  	 * @return null or the implementation of {@link CtScannerListener}, which is registered to listen for enter/exit of nodes during scanning of the AST
58  	 */
59  	public CtScannerListener getListener() {
60  		return listener;
61  	}
62  
63  	/**
64  	 * @param listener the implementation of {@link CtScannerListener}, which will be called back when entering/exiting
65  	 * odes during scanning.
66  	 * @return this to support fluent API
67  	 */
68  	public EarlyTerminatingScanner<T> setListener(CtScannerListener listener) {
69  		this.listener = listener;
70  		return this;
71  	}
72  
73  	@Override
74  	public void scan(CtRole role, Collection<? extends CtElement> elements) {
75  		if (isTerminated() || elements == null) {
76  			return;
77  		}
78  		// we use defensive copy so as to be able to change the class while scanning
79  		// otherwise one gets a ConcurrentModificationException
80  		for (CtElement e : new ArrayList<>(elements)) {
81  			scan(role, e);
82  			if (isTerminated()) {
83  				return;
84  			}
85  		}
86  	}
87  
88  	@Override
89  	public void scan(CtRole role, Map<String, ? extends CtElement> elements) {
90  		if (isTerminated() || elements == null) {
91  			return;
92  		}
93  		for (CtElement obj : elements.values()) {
94  			scan(role, obj);
95  			if (isTerminated()) {
96  				return;
97  			}
98  		}
99  	}
100 
101 	@Override
102 	public void scan(CtRole role, CtElement element) {
103 		scannedRole = role;
104 		super.scan(role, element);
105 	}
106 
107 	/*
108 	 * we cannot override scan(CtRole role, CtElement element) directly
109 	 * because some implementations needs scan(CtElement element), which must be called too
110 	 */
111 	@Override
112 	public void scan(CtElement element) {
113 		if (element == null || isTerminated()) {
114 			return;
115 		}
116 		if (listener == null) {
117 			//the listener is not defined
118 			//visit this element and may be children
119 			doScan(scannedRole, element, ScanningMode.NORMAL);
120 		} else {
121 			//the listener is defined, call it's enter method first
122 			ScanningMode mode = listener.enter(scannedRole, element);
123 			if (mode != ScanningMode.SKIP_ALL) {
124 				//the listener decided to visit this element and may be children
125 				doScan(scannedRole, element, mode);
126 				//then call exit, only if enter returned true
127 				listener.exit(scannedRole, element);
128 			} //else the listener decided to skip this element and all children. Do not call exit.
129 		}
130 	}
131 
132 	/**
133 	 * This method is called ONLY when the listener decides that the current element and children should be visited.
134 	 * Subclasses can override it to react accordingly.
135 	 */
136 	protected void doScan(CtRole role, CtElement element, ScanningMode mode) {
137 		//send input to output
138 		if (mode.visitElement) {
139 			onElement(role, element);
140 		}
141 		if (mode.visitChildren) {
142 			//do not call scan(CtElement) nor scan(CtRole, CtElement), because they would cause StackOverflowError
143 			element.accept(this);
144 		}
145 	}
146 
147 	@Override
148 	public void visitCtCompilationUnit(CtCompilationUnit compilationUnit) {
149 		if (isVisitCompilationUnitContent()) {
150 			enter(compilationUnit);
151 			scan(CtRole.COMMENT, compilationUnit.getComments());
152 			scan(CtRole.ANNOTATION, compilationUnit.getAnnotations());
153 			scan(CtRole.PACKAGE_DECLARATION, compilationUnit.getPackageDeclaration());
154 			scan(CtRole.DECLARED_IMPORT, compilationUnit.getImports());
155 			//visit directly the module (instead of reference only)
156 			scan(CtRole.DECLARED_MODULE, compilationUnit.getDeclaredModule());
157 			//visit directly the types (instead of references only)
158 			scan(CtRole.DECLARED_TYPE, compilationUnit.getDeclaredTypes());
159 			exit(compilationUnit);
160 		} else {
161 			super.visitCtCompilationUnit(compilationUnit);
162 		}
163 	}
164 
165 	/**
166 	 * Called for each scanned element. The call of this method is influenced by {@link ScanningMode} defined by {@link CtScannerListener}
167 	 * @param role a role of `element` in parent
168 	 * @param element a scanned element
169 	 */
170 	protected void onElement(CtRole role, CtElement element) {
171 	}
172 
173 	@Override
174 	public void scan(CtRole role, Object o) {
175 		if (isTerminated() || o == null) {
176 			return;
177 		}
178 		if (o instanceof CtElement) {
179 			scan(role, (CtElement) o);
180 		} else if (o instanceof Collection<?>) {
181 			scan(role, (Collection<? extends CtElement>) o);
182 		} else if (o instanceof Map<?, ?>) {
183 			for (Object obj : ((Map) o).values()) {
184 				scan(role, obj);
185 				if (isTerminated()) {
186 					return;
187 				}
188 			}
189 		}
190 	}
191 
192 	/**
193 	 * @return true if types and modules are visited. false if only their references are visited. false is default
194 	 */
195 	public boolean isVisitCompilationUnitContent() {
196 		return visitCompilationUnitContent;
197 	}
198 
199 	/**
200 	 * @param visitCompilationUnitContent use true if types and modules have to be visited. false if only their references have to be visited.
201 	 * false is default
202 	 */
203 	public EarlyTerminatingScanner<T> setVisitCompilationUnitContent(boolean visitCompilationUnitContent) {
204 		this.visitCompilationUnitContent = visitCompilationUnitContent;
205 		return this;
206 	}
207 }